/////////////////////////////////////////////////////////////
//
// pgAdmin 4 - PostgreSQL Tools
//
// Copyright (C) 2013 - 2021, The pgAdmin Development Team
// This software is released under the PostgreSQL Licence
//
//////////////////////////////////////////////////////////////

define('pgadmin.node.exclusion_constraint', [
  'sources/gettext', 'sources/url_for', 'jquery', 'underscore', 'backbone',
  'sources/pgadmin', 'pgadmin.browser', 'alertify', 'pgadmin.backform',
  'pgadmin.backgrid', 'pgadmin.browser.collection',
], function(
  gettext, url_for, $, _, Backbone, pgAdmin, pgBrowser, Alertify, Backform,
  Backgrid
) {

  var ExclusionConstraintColumnModel = pgBrowser.Node.Model.extend({
    defaults: {
      column: undefined,
      is_exp: false,
      oper_class: undefined,
      order: false,
      nulls_order: false,
      operator:undefined,
      col_type:undefined,
      is_sort_nulls_applicable: true,
    },
    toJSON: function () {
      var d = pgBrowser.Node.Model.prototype.toJSON.apply(this, arguments);
      delete d.col_type;
      return d;
    },
    schema: [{
      id: 'column', label: gettext('Col/Exp'), type:'text', editable: false,
      cell:'string',
    },{
      id: 'is_exp', label: '', type:'boolean', editable: false,
      cell: Backgrid.StringCell.extend({
        formatter: {
          fromRaw: function (rawValue) {
            return rawValue ? 'E' : 'C';
          },
          toRaw: function (val) {
            return val;
          },
        },
      }), visible: false,
    },{
      id: 'oper_class', label: gettext('Operator class'), type:'text',
      node: 'table', url: 'get_oper_class', first_empty: true,
      editable: function(m) {
        if (m instanceof Backbone.Collection) {
          return true;
        } else if ((_.has(m.collection, 'handler') &&
          !_.isUndefined(m.collection.handler) &&
            !_.isUndefined(m.collection.handler.get('oid')))) {
          return false;
        } else if (_.has(m.collection, 'handler') &&
              !_.isUndefined(m.collection.handler) &&
                !_.isUndefined(m.collection.handler.get('amname')) &&
                m.collection.handler.get('amname') != 'btree') {
          // Disable if access method is not btree
          return false;
        }
        return true;
      },
      select2: {
        allowClear: true, width: 'style', tags: true,
        placeholder: gettext('Select the operator class'),
      }, cell: Backgrid.Extension.Select2Cell.extend({
        initialize: function () {
          Backgrid.Extension.Select2Cell.prototype.initialize.apply(this, arguments);

          var self = this,
            url = self.column.get('url') || self.defaults.url,
            m = self.model,
            indextype = self.model.collection.handler.get('amname');

          if (url && (indextype == 'btree' || _.isUndefined(indextype) ||
            _.isNull(indextype) || indextype == '')) {
            // Set sort_order and nulls to true if access method is btree
            setTimeout(function() {
              m.set('order', true);
              m.set('nulls_order', true);
            }, 10);

            var node = this.column.get('schema_node'),
              eventHandler = m.top || m,
              node_info = this.column.get('node_info'),
              full_url = node.generate_url.apply(
                node, [
                  null, url, this.column.get('node_data'),
                  this.column.get('url_with_id') || false, node_info,
                ]),
              data = [];

            indextype = 'btree';

            if (this.column.get('version_compatible')) {
              eventHandler.trigger('pgadmin:view:fetching', m, self.column);
              $.ajax({
                async: false,
                data : {indextype:indextype},
                url: full_url,
              })
                .done(function(res) {
                  data = res.data;
                  self.column.set('options', data);
                })
                .fail(function() {
                  eventHandler.trigger('pgadmin:view:fetch:error', m, self.column);
                });
              eventHandler.trigger('pgadmin:view:fetched', m, self.column);
            }
          } else {
            self.column.set('options', []);
          }
        },
      }),
    },{
      id: 'order', label: gettext('DESC'), type: 'switch',
      options: {
        onText: 'ASC',
        offText: 'DESC',
      },
      editable: 'isEditable',
    },{
      id: 'nulls_order', label: gettext('NULLs order'), type:'switch',
      options: {
        onText: 'FIRST',
        offText: 'LAST',
      },
      editable: 'isEditable',
    },{
      id: 'operator', label: gettext('Operator'), type: 'text',
      node: 'table', url: 'get_operator',
      editable: function(m) {
        if (m instanceof Backbone.Collection) {
          return true;
        }
        if ((_.has(m.collection, 'handler') &&
          !_.isUndefined(m.collection.handler) &&
            !_.isUndefined(m.collection.handler.get('oid')))) {
          return false;
        }
        return true;
      },
      select2: {
        allowClear: false, width: 'style',
      }, cell: Backgrid.Extension.Select2Cell.extend({
        initialize: function () {
          Backgrid.Extension.Select2Cell.prototype.initialize.apply(this, arguments);

          var self = this,
            url = self.column.get('url') || self.defaults.url,
            m = self.model,
            col_type = self.model.get('col_type');

          self.column.set('options', []);

          if (url) {
            var node = this.column.get('schema_node'),
              eventHandler = m.top || m,
              node_info = this.column.get('node_info'),
              full_url = node.generate_url.apply(
                node, [
                  null, url, this.column.get('node_data'),
                  this.column.get('url_with_id') || false, node_info,
                ]),
              data = [];

            if (this.column.get('version_compatible')) {
              eventHandler.trigger('pgadmin:view:fetching', m, self.column);
              $.ajax({
                async: false,
                data : {col_type:col_type},
                url: full_url,
              })
                .done(function(res) {
                  data = res.data;
                  self.column.set('options', data);
                })
                .fail(function() {
                  eventHandler.trigger('pgadmin:view:fetch:error', m, self.column);
                });
              eventHandler.trigger('pgadmin:view:fetched', m, self.column);
            }
          }
        },
      }),
    },
    ],
    isEditable: function(m) {
      if (m instanceof Backbone.Collection) {
        return true;
      } else if ((_.has(m.collection, 'handler') &&
        !_.isUndefined(m.collection.handler) &&
        !_.isUndefined(m.collection.handler.get('oid')))) {
        return false;
      } else if (m.top.get('amname') === 'btree') {
        m.set('is_sort_nulls_applicable', true);
        return true;
      } else {
        m.set('is_sort_nulls_applicable', false);
        return false;
      }
    },
    validate: function() {
      this.errorModel.clear();
      var operator = this.get('operator'),
        column_name = this.get('column'),
        is_exp = this.get('is_exp');
      if (_.isUndefined(operator) || _.isNull(operator)) {
        var msg = gettext('Please specify operator for column: ') + column_name;
        if(is_exp) msg = gettext('Please specify operator for expression: ') + column_name;
        this.errorModel.set('operator', msg);
        return msg;
      }
      return null;
    },
  });

  var ExclusionConstraintColumnControl =  Backform.ExclusionConstraintColumnControl =
    Backform.UniqueColCollectionControl.extend({

      initialize: function() {
        Backform.UniqueColCollectionControl.prototype.initialize.apply(
          this, arguments
        );

        var self = this,
          node = 'exclusion_constraint',
          headerSchema = [{
            id: 'is_exp', label: gettext('Is expression ?'), type: 'switch',
            control: 'switch', controlLabelClassName: 'control-label pg-el-sm-4 pg-el-12',
            controlsClassName: 'pgadmin-controls pg-el-sm-6 pg-el-12',
          },{
            id: 'column', label: gettext('Column'), type:'text',
            controlLabelClassName: 'control-label pg-el-sm-4 pg-el-12',
            controlsClassName: 'pgadmin-controls pg-el-sm-6 pg-el-12',
            node: 'column', deps: ['is_exp'],
            control: Backform.NodeListByNameControl.extend({
              initialize: function() {
                // Here we will decide if we need to call URL
                // Or fetch the data from parent columns collection
                if(self.model.handler) {
                  Backform.Select2Control.prototype.initialize.apply(this, arguments);
                  // Do not listen for any event(s) for existing constraint.
                  if (_.isUndefined(self.model.get('oid'))) {
                    var tableCols = self.model.top.get('columns');
                    this.listenTo(tableCols, 'remove' , this.removeColumn);
                    this.listenTo(tableCols, 'change:name', this.resetColOptions);
                    this.listenTo(tableCols, 'change:cltype', this.resetColOptions);
                  }
                  this.custom_options();
                } else {
                  Backform.NodeListByNameControl.prototype.initialize.apply(this, arguments);
                }
              },
              removeColumn: function () {
                var that = this;
                setTimeout(function   () {
                  that.custom_options();
                  that.render.apply(that);
                }, 50);
              },
              resetColOptions: function(m) {
                var that = this;

                if (m.previous('name') ==  self.headerData.get('column')) {
                  /*
                   * Table column name has changed so update
                   * column name in exclusion constraint as well.
                   */
                  self.headerData.set(
                    {'column': m.get('name')});
                  self.headerDataChanged();
                }

                setTimeout(function () {
                  that.custom_options();
                  that.render.apply(that);
                }, 50);
              },
              custom_options: function() {
                // We will add all the columns entered by user in table model
                var columns = self.model.top.get('columns'),
                  added_columns_from_tables = [],
                  col_types = [];

                if (columns.length > 0) {
                  _.each(columns.models, function(m) {
                    var col = m.get('name');
                    if(!_.isUndefined(col) && !_.isNull(col)) {
                      added_columns_from_tables.push({
                        label: col, value: col, image:'icon-column',
                      });
                      col_types.push({name:col, type:m.get('cltype')});
                    }
                  });
                }
                // Set the values in to options so that user can select
                this.field.set('options', added_columns_from_tables);
                self.field.set('col_types', col_types);
              },
              remove: function () {
                if(self.model.handler) {
                  var tableCols = self.model.top.get('columns');
                  this.stopListening(tableCols, 'remove' , this.removeColumn);
                  this.stopListening(tableCols, 'change:name' , this.resetColOptions);
                  this.stopListening(tableCols, 'change:cltype' , this.resetColOptions);

                  Backform.Select2Control.prototype.remove.apply(this, arguments);

                } else {
                  Backform.NodeListByNameControl.prototype.remove.apply(this, arguments);
                }
              },
              template: _.template([
                '<span class="<%=controlLabelClassName%>"><%=label%></span>',
                '<div class="<%=controlsClassName%> <%=extraClasses.join(\' \')%>">',
                '  <select class="pgadmin-node-select form-control" name="<%=name%>" style="width:100%;" value="<%-value%>" <%=disabled ? "disabled" : ""%> <%=required ? "required" : ""%> >',
                '    <% for (var i=0; i < options.length; i++) { %>',
                '    <% var option = options[i]; %>',
                '    <option <% if (option.image) { %> data-image=<%= option.image %> <% } %> value=<%= formatter.fromRaw(option.value) %> <%=option.value === rawValue ? "selected=\'selected\'" : "" %>><%-option.label%></option>',
                '    <% } %>',
                '  </select>',
                '</div>'].join('\n')),
            }),
            transform: function(rows) {
              // This will only get called in case of NodeListByNameControl.

              var that = this,
                schema_node = that.field.get('schema_node'),
                res = [],
                col_types = [],
                filter = that.field.get('filter') || function() { return true; };

              filter = filter.bind(that);

              _.each(rows, function(r) {
                if (filter(r)) {
                  var l = (_.isFunction(schema_node['node_label']) ?
                      (schema_node['node_label']).apply(schema_node, [r, that.model, that]) :
                      r.label),
                    image = (_.isFunction(schema_node['node_image']) ?
                      (schema_node['node_image']).apply(
                        schema_node, [r, that.model, that]
                      ) :
                      (schema_node['node_image'] || ('icon-' + schema_node.type)));
                  res.push({
                    'value': r.label,
                    'image': image,
                    'label': l,
                  });
                  col_types.push({name:r.label, type:r.datatype});
                }
              });
              self.field.set('col_types', col_types);
              return res;
            },
            canAdd: function(m) {
              return !((_.has(m, 'handler') &&
                !_.isUndefined(m.handler) &&
                  !_.isUndefined(m.get('oid'))) || (_.isFunction(m.isNew) && !m.isNew()));
            },
            select2: {
              allowClear: false, width: 'style',
              placeholder: gettext('Select column'),
            }, first_empty: !self.model.isNew(),
            readonly: function() {
              return !_.isUndefined(self.model.get('oid'));
            },
            disabled: function(m) {
              return m.get('is_exp');
            },
          },{
            id: 'exp', label: gettext('Expression'), type: 'text',
            editable: true, deps: ['is_exp'],
            controlLabelClassName: 'control-label pg-el-sm-4 pg-el-12',
            controlsClassName: 'pgadmin-controls pg-el-sm-6 pg-el-12',
            disabled: function(m) {
              return !m.get('is_exp');
            },
          }],
          headerDefaults = {is_exp: false, column: null, exp: null},

          gridCols = ['is_exp', 'column', 'oper_class', 'order', 'nulls_order', 'operator'];

        self.headerData = new (Backbone.Model.extend({
          defaults: headerDefaults,
          schema: headerSchema,
        }))({});

        var headerGroups = Backform.generateViewSchema(
            self.field.get('node_info'), self.headerData, 'create',
            node, self.field.get('node_data')
          ),
          fields = [];

        _.each(headerGroups, function(o) {
          fields = fields.concat(o.fields);
        });

        self.headerFields = new Backform.Fields(fields);
        self.gridSchema = Backform.generateGridColumnsFromModel(
          self.field.get('node_info'), self.field.get('model'), 'edit', gridCols, self.field.get('schema_node')
        );

        self.controls = [];
        self.listenTo(self.headerData, 'change', self.headerDataChanged);
        self.listenTo(self.headerData, 'select2', self.headerDataChanged);
        self.listenTo(self.collection, 'add', self.onAddorRemoveColumns);
        self.listenTo(self.collection, 'remove', self.onAddorRemoveColumns);
      },

      generateHeader: function(data) {
        var isNew = _.isUndefined(this.model.get('oid'));
        var header = [
          '<div class="subnode-header-form '+ (isNew ? '' : 'd-none') +'">',
          ' <div>',
          '  <div header="is_exp"></div>',
          '  <div header="column"></div>',
          '  <div header="exp"></div>',
          ' </div>',
          '</div>'].join('\n');

        var self = this,
          headerTmpl = _.template(header),
          $header = $(headerTmpl(data)),
          controls = this.controls;

        this.headerFields.each(function(field) {
          var control = new (field.get('control'))({
            field: field,
            model: self.headerData,
          });

          $header.find('div[header="' + field.get('name') + '"]').append(
            control.render().$el
          );

          controls.push(control);
        });

        // We should not show in properties mode
        if (data.mode == 'properties') {
          $header.html('');
        }

        self.$header = $header;

        return $header;
      },

      events: _.extend(
        {}, Backform.UniqueColCollectionControl.prototype.events,
        {'click button.add': 'addColumns'}
      ),

      showGridControl: function(data) {
        var self = this,
          titleTmpl = _.template([
            '<div class="subnode-header">',
            '  <label class="control-label pg-el-sm-10"><%-label%></label>',
            '  <button class="btn btn-sm-sq btn-primary-icon add fa fa-plus" <%=canAdd ? "" : "disabled=\'disabled\'"%> title="' + gettext('Add new row') + '"></button>',
            '</div>'].join('\n')),
          $gridBody =
          $('<div class=\'pgadmin-control-group backgrid form-group col-12 object subnode\'></div>').append(
            // Append titleTmpl only if create/edit mode
            data.mode !== 'properties' ? titleTmpl({label: data.label, canAdd: data.canAdd}) : ''
          );

        $gridBody.append(self.generateHeader(data));

        var gridColumns = _.clone(this.gridSchema.columns);

        // Insert Delete Cell into Grid
        if (data.disabled == false && data.canDelete) {
          gridColumns.unshift({
            name: 'pg-backform-delete', label: '',
            cell: Backgrid.Extension.DeleteCell,
            editable: false, cell_priority: -1,
          });
        }

        if (self.grid) {
          self.grid.remove();
          self.grid = null;
        }
        // Initialize a new Grid instance
        var grid = self.grid = new Backgrid.Grid({
          columns: gridColumns,
          collection: self.collection,
          className: 'backgrid table-bordered table-noouter-border table-hover',
        });
        self.$grid = grid.render().$el;

        $gridBody.append(self.$grid);

        setTimeout(function() {
          self.headerData.set({
            'column': self.$header.find(
              'div[header="column"] select'
            ).val(),
          }, {silent:true}
          );
        }, 10);

        // Remove unwanted class from grid to display it properly
        if(data.mode === 'properties')
          $gridBody.find('.subnode-header-form').removeClass('subnode-header-form');

        // Render node grid
        self.$gridBody = $gridBody;
        return $gridBody;
      },

      headerDataChanged: function() {
        var self = this, val,
          data = this.headerData.toJSON(),
          inSelected = false,
          checkVars = ['column'];

        if (!self.$header) {
          return;
        }

        if (self.control_data.canAdd) {
          if(data['is_exp']) {
            inSelected = false;
          } else {
            self.collection.each(function(m) {
              if (!inSelected) {
                _.each(checkVars, function(v) {
                  if (!inSelected) {
                    val = m.get(v);
                    inSelected = ((
                      (_.isUndefined(val) || _.isNull(val)) &&
                        (_.isUndefined(data[v]) || _.isNull(data[v]))
                    ) ||
                        (val == data[v]));
                  }
                });
              }
            });
          }
        }
        else {
          inSelected = true;
        }

        self.$gridBody.find('button.add').prop('disabled', inSelected);
      },

      addColumns: function(ev) {
        ev.preventDefault();
        var self = this;
        let newHeaderData = {
          is_exp: self.headerData.get('is_exp'),
          column: self.headerData.get('is_exp') ? self.headerData.get('exp') : self.headerData.get('column'),
        };

        if (newHeaderData.column && newHeaderData.column != '') {
          var coll = self.model.get(self.field.get('name')),
            m = new (self.field.get('model'))(
              newHeaderData, {
                silent: true, top: self.model.top,
                collection: coll, handler: coll,
              }),
            col_types =self.field.get('col_types') || [];

          if(!m.get('is_exp')) {
            for(var i=0; i < col_types.length; i++) {
              var col_type = col_types[i];
              if (col_type['name'] ==  m.get('column')) {
                m.set({'col_type':col_type['type']});
                break;
              }
            }
          }

          coll.add(m);

          var idx = coll.indexOf(m);

          // idx may not be always > -1 because our UniqueColCollection may
          // remove 'm' if duplicate value found.
          if (idx > -1) {
            self.$grid.find('.new').removeClass('new');

            var newRow = self.grid.body.rows[idx].$el;

            newRow.addClass('new');
            $(newRow).pgMakeVisible('backform-tab');
          } else {
            //delete m;
          }
        }

        return false;
      },

      onAddorRemoveColumns: function() {
        var self = this;

        // Wait for collection to be updated before checking for the button to be
        // enabled, or not.
        setTimeout(function() {
          self.collection.trigger('pgadmin:columns:updated', self.collection);
          self.headerDataChanged();
        }, 10);
      },

      remove: function() {
        /*
         * Stop listening the events registered by this control.
         */
        this.stopListening(this.headerData, 'change', this.headerDataChanged);
        this.listenTo(this.headerData, 'select2', this.headerDataChanged);
        this.listenTo(this.collection, 'remove', this.onAddorRemoveColumns);

        // Remove header controls.
        _.each(this.controls, function(controls) {
          controls.remove();
        });

        ExclusionConstraintColumnControl.__super__.remove.apply(this, arguments);

        // Remove the header model
        delete (this.headerData);

      },
    });

  // Extend the browser's node class for exclusion constraint node
  if (!pgBrowser.Nodes['exclusion_constraint']) {
    pgAdmin.Browser.Nodes['exclusion_constraint'] = pgBrowser.Node.extend({
      type: 'exclusion_constraint',
      label: gettext('Exclusion constraint'),
      collection_type: 'coll-constraints',
      sqlAlterHelp: 'ddl-alter.html',
      sqlCreateHelp: 'ddl-constraints.html',
      dialogHelp: url_for('help.static', {'filename': 'exclusion_constraint_dialog.html'}),
      hasSQL: true,
      parent_type: ['table','partition'],
      canDrop: true,
      canDropCascade: true,
      hasDepends: true,
      hasStatistics: true,
      statsPrettifyFields: [gettext('Index size')],
      url_jump_after_node: 'schema',
      Init: function() {
        /* Avoid multiple registration of menus */
        if (this.initialized)
          return;

        this.initialized = true;

        pgBrowser.add_menus([{
          name: 'create_exclusion_constraint_on_coll', node: 'coll-constraints', module: this,
          applies: ['object', 'context'], callback: 'show_obj_properties',
          category: 'create', priority: 4, label: gettext('Exclusion constraint...'),
          icon: 'wcTabIcon icon-exclusion_constraint', data: {action: 'create', check: true},
          enable: 'canCreate',
        }]);
      },
      is_not_valid: function(node) {
        return (node && !node.valid);
      },
      // Define the model for exclusion constraint node
      model: pgAdmin.Browser.Node.Model.extend({
        idAttribute: 'oid',

        defaults: {
          name: undefined,
          oid: undefined,
          is_sys_obj: undefined,
          comment: undefined,
          spcname: undefined,
          amname: 'gist',
          fillfactor: undefined,
          condeferrable: undefined,
          condeferred: undefined,
          columns: [],
          include: [],
        },

        // Define the schema for the exclusion constraint node
        schema: [{
          id: 'name', label: gettext('Name'), type: 'text',
          mode: ['properties', 'create', 'edit'], editable: true,
        },{
          id: 'oid', label: gettext('OID'), cell: 'string',
          type: 'text' , mode: ['properties'],
        },{
          id: 'is_sys_obj', label: gettext('System exclusion constraint?'),
          cell:'boolean', type: 'switch', mode: ['properties'],
        },{
          id: 'comment', label: gettext('Comment'), cell: 'string',
          type: 'multiline', mode: ['properties', 'create', 'edit'],
          deps:['name'], disabled:function(m) {
            var name = m.get('name');
            if (!(name && name != '')) {
              setTimeout(function(){
                if(m.get('comment') && m.get('comment') !== '')
                  m.set('comment', null);
              },10);
              return true;
            } else {
              return false;
            }
          },
        },{
          id: 'spcname', label: gettext('Tablespace'),
          type: 'text', group: gettext('Definition'),
          control: 'node-list-by-name', node: 'tablespace',
          select2:{allowClear:false},
          filter: function(m) {
            // Don't show pg_global tablespace in selection.
            return (m.label !== 'pg_global');
          },
        },{
          id: 'amname', label: gettext('Access method'),
          type: 'text', group: gettext('Definition'),
          url:'get_access_methods', node: 'table',
          control: Backform.NodeAjaxOptionsControl.extend({
            // When access method changes we need to clear columns collection
            onChange: function() {
              Backform.NodeAjaxOptionsControl.prototype.onChange.apply(this, arguments);
              var self = this,
                // current access method
                current_am = self.model.get('amname'),
                // previous access method
                previous_am = self.model.previous('amname'),
                column_collection = self.model.get('columns');

              if (column_collection.length > 0 && current_am != previous_am) {
                var msg = gettext('Changing access method will clear columns collection');
                Alertify.confirm(msg, function () {
                  // User clicks Ok, lets clear collection.
                  column_collection.each(function(m) {
                    /*
                     * Our datamodel do not support collection reset method.
                     * So remove model one by one.
                     */
                    column_collection.remove(m);
                  });
                  setTimeout(function() {
                    column_collection.trigger('pgadmin:columns:updated', column_collection);
                  }, 10);

                }, function() {
                  // User clicks Cancel set previous value again in combo box
                  setTimeout(function(){
                    self.model.set('amname', previous_am);
                  }, 10);
                });
              }
            },
          }),
          select2:{allowClear:true},
          readonly: 'isReadonly',
        },{
          id: 'fillfactor', label: gettext('Fill factor'),
          type: 'int', group: gettext('Definition'), allowNull: true,
        },{
          id: 'condeferrable', label: gettext('Deferrable?'),
          type: 'switch', group: gettext('Definition'), deps: ['index'],
          readonly: 'isReadonly',
        },{
          id: 'condeferred', label: gettext('Deferred?'),
          type: 'switch', group: gettext('Definition'),
          deps: ['condeferrable'],
          disabled: function(m) {
            // Disable if condeferred is false or unselected.
            if(m.get('condeferrable') == true) {
              return false;
            } else {
              setTimeout(function(){
                if(m.get('condeferred'))
                  m.set('condeferred', false);
              },10);
              return true;
            }
          },
          readonly: function(m) {
            if((_.has(m, 'handler') &&
              !_.isUndefined(m.handler) &&
                !_.isUndefined(m.get('oid'))) || (_.isFunction(m.isNew) && !m.isNew())) {
              return true;
            }
          },
        },{
          id: 'indconstraint', label: gettext('Constraint'), cell: 'string',
          type: 'multiline', mode: ['create', 'edit', 'properties'], editable: false,
          group: gettext('Definition'), readonly: 'isReadonly',
        },{
          id: 'columns', label: gettext('Columns/Expressions'),
          type: 'collection', group: gettext('Columns'),
          deps:['amname'], canDelete: true, editable: false,
          canAdd: function(m) {
            // We can't update columns of existing exclusion constraint.
            return !((_.has(m, 'handler') &&
              !_.isUndefined(m.handler) &&
                !_.isUndefined(m.get('oid'))) || (_.isFunction(m.isNew) && !m.isNew()));
          },
          control: ExclusionConstraintColumnControl,
          model: ExclusionConstraintColumnModel,
          readonly: 'isReadonly',
          cell: Backgrid.StringCell.extend({
            initialize: function() {
              Backgrid.StringCell.prototype.initialize.apply(this, arguments);
              var self = this;
              // Do not listen for any event(s) for existing constraint.
              if (_.isUndefined(self.model.get('oid'))) {
                var tableCols = self.model.top.get('columns');

                self.listenTo(tableCols, 'remove' , self.removeColumn);
                self.listenTo(tableCols, 'change:name', self.resetColOptions);
                self.listenTo(tableCols, 'change:cltype', self.removeColumnWithType);
              }

              this.model.get('columns').on('pgadmin:columns:updated', function() {
                self.render.apply(self);
              });
            },
            removeColumnWithType: function(m){
              var self = this,
                cols = self.model.get('columns'),
                removedCols = cols.where(
                  {col_type: m.previous('cltype')}
                );

              cols.remove(removedCols);
              setTimeout(function () {
                self.render();
              }, 10);

              setTimeout(function () {
                var constraints = self.model.top.get('exclude_constraint');
                var removed = [];
                constraints.each(function(constraint) {
                  if (constraint.get('columns').length == 0) {
                    removed.push(constraint);
                  }
                });
                constraints.remove(removed);
              },100);
            },
            removeColumn: function(m){
              var self = this,
                removedCols = self.model.get('columns').where(
                  {column: m.get('name')}
                );

              self.model.get('columns').remove(removedCols);
              setTimeout(function () {
                self.render();
              }, 10);

              setTimeout(function () {
                var constraints = self.model.top.get('exclude_constraint');
                var removed = [];
                constraints.each(function(constraint) {
                  if (constraint.get('columns').length == 0) {
                    removed.push(constraint);
                  }
                });
                constraints.remove(removed);
              },100);
            },
            resetColOptions : function(m) {
              var self = this,
                updatedCols = self.model.get('columns').where(
                  {'column': m.previous('name')}
                );

              if (updatedCols.length > 0) {
                /*
                 * Table column name has changed so update
                 * column name in foreign key as well.
                 */
                updatedCols[0].set(
                  {'column': m.get('name')});
              }

              setTimeout(function () { self.render(); }, 10);
            },
            formatter: {
              fromRaw: function (rawValue) {
                return rawValue.pluck('column').toString();
              },
              toRaw: function (val) {
                return val;
              },
            },
            render: function() {
              return Backgrid.StringCell.prototype.render.apply(this, arguments);
            },
            remove: function() {
              var tableCols = this.model.top.get('columns'),
                cols = this.model.get('columns');
              if (cols) {
                cols.off('pgadmin:columns:updated');
              }

              this.stopListening(tableCols, 'remove' , self.removeColumn);
              this.stopListening(tableCols, 'change:name' , self.resetColOptions);
              this.stopListening(tableCols, 'change:cltype' , self.removeColumnWithType);

              Backgrid.StringCell.prototype.remove.apply(this, arguments);
            },
          }),
        },{
          id: 'include', label: gettext('Include columns'),
          type: 'array', group: gettext('Columns'),
          editable: false,
          canDelete: true, canAdd: true, mode: ['properties', 'create', 'edit'],
          visible: function(m) {
            /* In table properties, m.node_info is not available */
            m = m.top;
            if(!_.isUndefined(m.node_info) && !_.isUndefined(m.node_info.server)
              && !_.isUndefined(m.node_info.server.version) &&
              m.node_info.server.version >= 110000)
              return true;

            return false;
          },
          control: Backform.MultiSelectAjaxControl.extend({
            defaults: _.extend(
              {},
              Backform.NodeListByNameControl.prototype.defaults,
              {
                select2: {
                  allowClear: false,
                  width: 'style',
                  multiple: true,
                  placeholder: gettext('Select the column(s)'),
                },
              }
            ),
            initialize: function() {
              // Here we will decide if we need to call URL
              // Or fetch the data from parent columns collection
              var self = this;
              if(this.model.handler) {
                Backform.Select2Control.prototype.initialize.apply(this, arguments);
                // Do not listen for any event(s) for existing constraint.
                if (_.isUndefined(self.model.get('oid'))) {
                  var tableCols = self.model.top.get('columns');
                  self.listenTo(tableCols, 'remove' , self.resetColOptions);
                  self.listenTo(tableCols, 'change:name', self.resetColOptions);
                }

                self.custom_options();
              } else {
                Backform.MultiSelectAjaxControl.prototype.initialize.apply(this, arguments);
              }
            },
            resetColOptions: function() {
              var self = this;

              setTimeout(function () {
                self.custom_options();
                self.render.apply(self);
              }, 50);
            },
            custom_options: function() {
              // We will add all the columns entered by user in table model
              var columns = this.model.top.get('columns'),
                added_columns_from_tables = [];

              if (columns.length > 0) {
                _.each(columns.models, function(m) {
                  var col = m.get('name');
                  if(!_.isUndefined(col) && !_.isNull(col)) {
                    added_columns_from_tables.push(
                      {label: col, value: col, image:'icon-column'}
                    );
                  }
                });
              }
              // Set the values in to options so that user can select
              this.field.set('options', added_columns_from_tables);
            },
          }),
          deps: ['index'], node: 'column',
          disabled: function(m) {
            // Disable if index is selected.
            var index = m.get('index');
            if(_.isUndefined(index) || index == '') {
              return false;
            } else {
              var col = m.get('columns');
              col.reset();
              return true;
            }
          },
          readonly: function(m) {
            // If we are in table edit mode then
            if (_.has(m, 'top') && !_.isUndefined(m.top)
              && !m.top.isNew()) {
              // If OID is undefined then user is trying to add
              // new constraint which should be allowed for Unique
              return !_.isUndefined(m.get('oid'));
            }

            // We can't update columns of existing index constraint.
            return !m.isNew();
          },
        }],
        isReadonly: function(m) {
          return ((_.has(m, 'handler') &&
              !_.isUndefined(m.handler) &&
              !_.isUndefined(m.get('oid'))) || (_.isFunction(m.isNew) && !m.isNew()));
        },
        validate: function() {
          this.errorModel.clear();
          var columns = this.get('columns'),
            name = this.get('name'),
            msg;

          if ((_.isUndefined(name) || _.isNull(name) || name.length < 1)) {
            msg = gettext('Please specify name for exclusion constraint.');
            this.errorModel.set('name', msg);
            return msg;
          } else  if (
            (_.isUndefined(columns) || _.isNull(columns) || columns.length < 1)
          ) {
            msg = gettext('Please specify columns for exclusion constraint.');
            this.errorModel.set('columns', msg);
            return msg;
          }

          return null;
        },
      }),

      canCreate: function(itemData, item, data) {
        // If check is false then , we will allow create menu
        if (data && data.check == false)
          return true;

        var t = pgBrowser.tree, i = item, d = itemData, parents = [],
          immediate_parent_table_found = false,
          is_immediate_parent_table_partitioned = false;
        // To iterate over tree to check parent node
        while (i) {
          // If table is partitioned table then return false
          if (!immediate_parent_table_found && (d._type == 'table' || d._type == 'partition')) {
            immediate_parent_table_found = true;
            if ('is_partitioned' in d && d.is_partitioned) {
              is_immediate_parent_table_partitioned = true;
            }
          }

          // If it is schema then allow user to create table
          if (_.indexOf(['schema'], d._type) > -1)
            return !is_immediate_parent_table_partitioned;
          parents.push(d._type);
          i = t.hasParent(i) ? t.parent(i) : null;
          d = i ? t.itemData(i) : null;
        }
        // If node is under catalog then do not allow 'create' menu
        if (_.indexOf(parents, 'catalog') > -1) {
          return false;
        } else {
          return !is_immediate_parent_table_partitioned;
        }
      },
    });
  }

  return pgBrowser.Nodes['exclusion_constraint'];
});
